Spring Data是什么Spring Data是一个用于简化数据库访问,并支持云服务的开源框架。其主要目标是使得对数据的访问变得方便快捷,并支持map-reduce框架和云计算数据服务。 Spring Data 包含多个子项目:Commons - 提供共享的基础框架,适合各个子项目使用,支持跨数据库持久化JPA - 简化创建 JPA 数据访问层和跨存储的持久层功能Hadoop - 基于 Spring 的 Hadoop 作业配置和一个 POJO 编程模型的 MapReduce 作业Key-Value - 集成了 Redis 和 Riak ,提供多个常用场景下的简单封装Document - 集成文档数据库:CouchDB 和 MongoDB 并提供基本的配置映射和资料库支持Graph - 集成 Neo4j 提供强大的基于 POJO 的编程模型Graph Roo AddOn - Roo support for Neo4jJDBC Extensions - 支持 Oracle RAD、高级队列和高级数据类型Mapping - 基于 Grails 的提供对象映射框架,支持不同的数据库Examples - 示例程序、文档和图数据库Guidance - 高级文档Spring Data JPA是什么由Spring提供的一个用于简化JPA开发的框架nSpring Data JPA能干什么可以极大的简化JPA的写法,可以在几乎不用写实现的情况下,实现对数据的访问和操作。除了CRUD外,还包括如分页、排序等一些常用的功能。Spring Data JPA有什么主要来看看Spring Data JPA提供的接口,也是Spring Data JPA的核心概念:1:Repository:最顶层的接口,是一个空的接口,目的是为了统一所有Repository的类型,且能让组件扫描的时候自动识别。2:CrudRepository :是Repository的子接口,提供CRUD的功能3:PagingAndSortingRepository:是CrudRepository的子接口,添加分页和排序的功能4:JpaRepository:是PagingAndSortingRepository的子接口,增加了一些实用的功能,比如:批量操作等。5:JpaSpecificationExecutor:用来做负责查询的接口6:Specification:是Spring Data JPA提供的一个查询规范,要做复杂的查询,只需围绕这个规范来设置查询条件即可 HelloWorldn环境构建在Eclipse里面构建一个普通的Java工程,主要就是要加入一堆的jar包。1:首先去官网下载Spring Data Common 和 Spring Data JPA的包,把里面dist的jar包加入到工程中,这里是spring-data-commons-1.5.0.RELEASE.jar和spring-data-jpa-1.3.2.RELEASE.jar2:把Spring3.2.3的jar包添加到工程中3:JPA的实现选用的是Hibernate4.2.0,总共还需要额外加入如下的jar:antlr-2.7.7.jar、aopalliance-1.0.jar、asm-3.2.jar、aspectjrt-1.7.1.jar、aspectjweaver-1.7.1.jar、commons-beanutils-1.8.3.jar、commons-codec-1.7.jar、commons-collections-3.2.1.jar、commons-dbcp-1.4.jar、commons-fileupload-1.2.2.jar、commons-io-2.4.jar、commons-lang3-3.1.jar、commons-logging-1.1.1.jar、commons-pool-1.6.jar、dom4j-1.6.1.jar、hibernate-commons-annotations-4.0.1.Final.jar、hibernate-core-4.2.0.Final.jar、hibernate-entitymanager-4.2.0.Final.jar、hibernate-jpa-2.0-api-1.0.1.Final.jar、javassist-3.15.0-GA.jar、jboss-logging-3.1.0.GA.jar、jboss-transaction-api_1.1_spec-1.0.0.Final.jar、mysql-connector-java-5.1.9.jar、slf4j-api-1.7.3.jarn实体对象,就是以前的实现方式@Entity@Table(name=”tbl_user”)public class UserModel {@Idprivate Integer uuid;private String name;private Integer age;//省略getter/setter}nDAO的接口public interface UserRepository extends JpaRepository{//空的,可以什么都不用写}无需提供实现,Spring Data JPA会为我们搞定一切n写个逻辑层的Service,其实就相当于DAO的客户端,用来测试@Service@Transactionalpublic class Client {@Autowiredprivate UserRepository ur; public void testAdd(UserModel um){ ur.save(um); } public static void main(String[] args) {ApplicationContext ctx = new ClassPathXmlApplicationContext(“applicationContext.xml”); Client c = (Client)ctx.getBean(“client”);UserModel um = new UserModel();um.setAge(1);um.setName(“张三”);um.setUuid(1); c.testAdd(um);} }n同样需要在Spring的配置文件中配置,基本跟使用注解的配置类似:<?xml version=”1.0” encoding=”UTF-8”?>org.gjt.mm.mysql.Driverjdbc:mysql://localhost:3306/cc?useUnicode=true&characterEncoding=UTF-8root 配置完成后,可以去运行Client测试一下了,当然数据库和表需要先准备好也可以在下面添加filter,形如: ##第二章:JpaRepository基本功能
JpaRepository的基本功能示范具体的看代码演示其中:Pageable接口的实现类是PageRequest,Page接口的实现类是PageImpl。示例如下:Page p = ur.findAll(new PageRequest(0,2,new Sort(new Order(Direction. DESC,"uuid"))));System. out.println("list="+p.getContent()); ##第三章:JpaRepository的查询
直接在接口中定义查询方法,如果是符合规范的,可以不用写实现,目前支持的关键字写法如下:   Spring Data JPA框架在进行方法名解析时,会先把方法名多余的前缀截取掉,比如 find、findBy、read、readBy、get、getBy,然后对剩下部分进行解析。假如创建如下的查询:findByUserDepUuid(),框架在解析该方法时,首先剔除 findBy,然后对剩下的属性进行解析,假设查询实体为Doc1:先判断 userDepUuid (根据 POJO 规范,首字母变为小写)是否为查询实体的一个属性,如果是,则表示根据该属性进行查询;如果没有该属性,继续第二步;2:从右往左截取第一个大写字母开头的字符串此处为Uuid),然后检查剩下的字符串是否为查询实体的一个属性,如果是,则表示根据该属性进行查询;如果没有该属性,则重复第二步,继续从右往左截取;最后假设user为查询实体的一个属性;3:接着处理剩下部分(DepUuid),先判断 user 所对应的类型是否有depUuid属性,如果有,则表示该方法最终是根据 “ Doc.user.depUuid” 的取值进行查询;否则继续按照步骤 2 的规则从右往左截取,最终表示根据 “Doc.user.dep.uuid” 的值进行查询。4:可能会存在一种特殊情况,比如 Doc包含一个 user 的属性,也有一个 userDep 属性,此时会存在混淆。可以明确在属性之间加上 "_" 以显式表达意图,比如 "findByUser_DepUuid()" 或者 "findByUserDep_uuid()"特殊的参数: 还可以直接在方法的参数上加入分页或排序的参数,比如:Page findByName(String name, Pageable pageable);List findByName(String name, Sort sort); 也可以使用JPA的NamedQueries,方法如下:1:在实体类上使用@NamedQuery,示例如下:@NamedQuery(name = "UserModel.findByAge",query = "select o from UserModel o where o.age >= ?1”)2:在自己实现的DAO的Repository接口里面定义一个同名的方法,示例如下:public List findByAge(int age);3:然后就可以使用了,Spring会先找是否有同名的NamedQuery,如果有,那么就不会按照接口定义的方法来解析。 使用@Query可以在自定义的查询方法上使用@Query来指定该方法要执行的查询语句,比如:@Query(“select o from UserModel o where o.uuid=?1”)public List findByUuidOrAge(int uuid);注意:1:方法的参数个数必须和@Query里面需要的参数个数一致2:如果是like,后面的参数需要前面或者后面加“%”,比如下面都对:@Query(“select o from UserModel o where o.name like ?1%”)public List findByUuidOrAge(String name); @Query(“select o from UserModel o where o.name like %?1”)public List findByUuidOrAge(String name); @Query(“select o from UserModel o where o.name like %?1%”)public List findByUuidOrAge(String name); 当然,这样在传递参数值的时候就可以不加‘%’了,当然加了也不会错 n还可以使用@Query来指定本地查询,只要设置nativeQuery为true,比如:@Query(value=”select * from tbl_user where name like %?1” ,nativeQuery=true)public List findByUuidOrAge(String name);注意:当前版本的本地查询不支持翻页和动态的排序 使用命名化参数,使用@Param即可,比如:@Query(value=”select o from UserModel o where o.name like %:nn”)public List findByUuidOrAge(@Param(“nn”) String name);同样支持更新类的Query语句,添加@Modifying即可,比如:@Modifying@Query(value=”update UserModel o set o.name=:newName where o.name like %:nn”)public int findByUuidOrAge(@Param(“nn”) String name,@Param(“newName”) String newName);注意:1:方法的返回值应该是int,表示更新语句所影响的行数2:在调用的地方必须加事务,没有事务不能正常执行 JpaRepository的查询功能创建查询的顺序Spring Data JPA 在为接口创建代理对象时,如果发现同时存在多种上述情况可用,它该优先采用哪种策略呢? 提供了 query-lookup-strategy 属性,用以指定查找的顺序。它有如下三个取值:1:create-if-not-found:如果方法通过@Query指定了查询语句,则使用该语句实现查询;如果没有,则查找是否定义了符合条件的命名查询,如果找到,则使用该命名查询;如果两者都没有找到,则通过解析方法名字来创建查询。这是 query-lookup-strategy 属性的默认值2:create:通过解析方法名字来创建查询。即使有符合的命名查询,或者方法通过 @Query指定的查询语句,都将会被忽略3:use-declared-query:如果方法通过@Query指定了查询语句,则使用该语句实现查询;如果没有,则查找是否定义了符合条件的命名查询,如果找到,则使用该命名查询;如果两者都没有找到,则抛出异常 ##第四章:客户化扩展JpaRepository 如果你不想暴露那么多的方法,可以自己订制自己的Repository,还可以在自己的Repository里面添加自己使用的公共方法当然更灵活的是自己写一个实现类,来实现自己需要的方法1:写一个与接口同名的类,加上后缀为Impl,这个在前面xml里面配置过,可以自动被扫描到。这个类不需要实现任何接口。2:在接口中加入自己需要的方法,比如:public Page
@PersistenceContext
private EntityManager em;
public Page
String hql = “select o.uuid,o.name from UserModel o where 1=1 and o.uuid=:uuid”;